



MICROCOPY RESOLUTION TEST CHART NATIONAL BUREAU OF STANDARDS-1963-A

# AD-A161 930

SCIENTIFIC AND TECHNICAL REPORT

FIFTH QUARTERLY STATUS REPORT

Prepared By

Stephen Y.H. Su Project Director Dept. of Computer Science State University of New York Binghamton, New York 13901 (607)-798-2296 (office) (607)-798-4802 (secretary)

A. Contractor's name and address:

The Research Foundation of State University of New York P.O. Box 9, Albany, New York 12201

- E. Contract No. DAABO7-82-K J05-6
- C. Date of Report November 3, 1983
- D. Title: The Fourth Quarterly Status Report for the project "Functional Testing of LSI/VLSI Based Systems with Measure of Fault Coverage"
- E. Period Covered: July 28 to October 27, 1983
- F. Description of Incomession Secontracked
- G. Spending: Actual Quarter Cost: \$ 41,837.00

Curulative Cost to date: \$132,627.15



88 058

This document has been approved for public release and sale; its distribution is unlimited.

### A FUNCTIONAL TESTING METHOD FOR MICKOPROCESSORS\*

by

Li Shen\*\*

Stephen Y.H. Su

Department of Computer Science
Thomas J. Watson School of Engineering,
Applied Science, and Technology
State University of New York
Binghamton, N.Y. 13901





This paper presents a method for functional testing of microprocessors.

First, we develop a control fault model at the RTL (Register Transfer Language)

level. Based on this model, we establish testing requirements for control

faults. We present three test procedures to verify the write and read sequences
and use the write and read sequences to test other instructions in a micro
processor. By utilizing k-out-of-m codes, we can use fewer tests to cover

more faults, thereby reducing the test generation time.



- \* This work is supported by the U.S. Army Communication Electronics Command under Research Contract No. DAAB 07-82-K-J056.
- \*\* Li Shen is now a visiting scholar in SUNY-Binghamton, from Institute of Computing Technology, Chinese Academy of Sciences, Ecijing, China.

### I. INTRODUCTION

The development of integrated circuit technology has resulted in a wide range of applications for microprocessors. Testing of microprocessors is a difficult problem because of the complexities of microprocessors. The problem is more serious for users due to lack of information on internal implementation of microprocessors and other VLSI chips. Over the past few years, several methods have been proposed to solve this problem. These testing techniques are essentially based on functional level [1-11].

A microprocessor is a type of complex sequential rachine. The current approach is to test microprocessors by instruction execution. Generally, before executing an instruction-under-test we have to write certain data into some registers, and after executing the instruction, read the contents of the registers. Therefore, if the write or the read instruction is faulty, we may not be able to test the instruction-under-test. To solve this problem, Thatte and Abraham [3] have to label instructions and define test order in detail before testing. However, they do not consider the partial execution of an instruction. So for instruction decoding fault  $I_j/I_j + I_k$ , it is assumed that instead of executing  $I_j$ , both instructions  $I_j$  and  $I_k$  are executed to completion. It is more general and practical to consider partial execution of an instruction under fault. Our fault model allows this.

Abraham and Parker [6] proposed a simplified fault model. First, one tests all internal registers, then executes all instruction and data manipulation functions.

In this paper, we consider the basic instructions for the write and read register functions as the kernel of a microprocessor. This kernel can be represented by a sequential machine. Based on the fault model, we use checking experiment

ではいいいないは、一方ではないから

to verify the kernel. Then we use the kernel for testing other instructions. The control fault model is established at the Register Transfer Language (RTL) level, since it is convenient to represent the instruction decoding faults and other control faults at such a level.

Section II presents a fault model for microprocessors, emphasizing the control fault model defined at the RTL level instead of the instruction level. In Section III, after examining most existing off-the-shelf microprocessors, we derive testing requirements based on different types of operations. In Section IV, we define the write and read sequences as the kernel of a microprocessor. Then Section V presents the verification of the write and read sequences. Section VI discusses the testing of control faults. Finally, conclusions are given in Section VII.

### II. FAULT MODEL

The functions of a microprocessor are mainly performed by instruction execution. The sequence of operations for an instruction can be described by RTL. We consider that an instruction consists of a series of RTL statements. The typical statement is defined as

(conditions): D+f (S<sub>1</sub>, S<sub>2</sub>, ..., S<sub>i</sub>, ...) where

D - destination

S<sub>4</sub> - Source

 $f(S_1, S_2, \ldots, S_1, \ldots)$  - operation

Destinations and sources may be internal registers of a microprocessor or external to the microprocessor (i.e., data bus, address bus, etc.). We are only concerned with those internal registers which are of interest to users, so we do not consider implied registers such as buffers. For example, data transfer from memory to memory can be described as  $DB_j \cdot DB_i$ , instead of Buffer  $DE_i$  followed by  $DE_j \cdot Buffer$ , where DE denotes the data bus which represents data input or output of memory, i,j denote different bus cycles,  $DE_i$  (read from memory) is ahead of  $DE_i$  (write into memory).

After examining most existing off-the-shelf microprocessors, e.g. Intel 8080 and 8086, Zilog 80 and 8000, Motorola 6800 and 68000, the RTL-like operations can be divided into two classes, transfer operations (class T, D+S), and arithmetic and logical operations (class A). Class A can be subdivided into six subclasses based on the combination of destinctions and sources as shown in Table 1, where the content of flag bits constitute a status register.

| Class | Type of Expression                                   | Operation                                                   |  |
|-------|------------------------------------------------------|-------------------------------------------------------------|--|
| A1    | D+f (D)                                              | EIT SET                                                     |  |
|       |                                                      | BIT RESET                                                   |  |
|       |                                                      | BIT COMPLEMENT                                              |  |
|       | ·                                                    | INCREMENT                                                   |  |
|       |                                                      | DECREMENT                                                   |  |
|       |                                                      | DECIMAL ADJUST                                              |  |
|       |                                                      | SHIFT                                                       |  |
|       |                                                      | ROTATE                                                      |  |
|       |                                                      | CO!PLEMENT                                                  |  |
|       |                                                      | NEGATE                                                      |  |
|       |                                                      | CLEAR                                                       |  |
| A2    | D+f(D,S)                                             | ADDITION                                                    |  |
|       |                                                      | ADDITION WITH CARRY                                         |  |
|       |                                                      | SUETRACTION                                                 |  |
|       |                                                      | SUBTRACTION WITH BORROW                                     |  |
| {     |                                                      | AND                                                         |  |
|       |                                                      | OR                                                          |  |
|       |                                                      | xor                                                         |  |
|       | D+f (S)                                              | EXTEND SIGN                                                 |  |
| A3    | Def(S <sub>1</sub> ,S <sub>2</sub> )                 | ADDITION, Des <sub>1</sub> +s <sub>2</sub>                  |  |
|       | D-f(D,S <sub>1</sub> ,S <sub>2</sub> )               | ADDITION, D+D+S <sub>1</sub> +S <sub>2</sub>                |  |
| A4    | D+f(S <sub>1</sub> ,S <sub>2</sub> ,S <sub>3</sub> ) | ADDITION, De-S <sub>1</sub> +S <sub>2</sub> +S <sub>3</sub> |  |
| AM    | D <sub>1</sub> ,D <sub>2</sub> +f(D <sub>1</sub> ,S) | MULTIPLY                                                    |  |
|       |                                                      | DIVIDE                                                      |  |
| AF    | Flags+f(S)                                           | BIT TEST                                                    |  |
|       | Flags+f(S <sub>1</sub> ,S <sub>2</sub> )             | CO) PARE                                                    |  |
|       | Flags+f(S <sub>1</sub> ,S <sub>2</sub> ,)            | Modifying flags for all                                     |  |
|       |                                                      | arithmetic and logical instructions                         |  |

Table 1. RTL-Like Operations

Note that the control operations in RTL control statements such as conditional branch are not listed because we can use RTL assignment statements with conditions and expand the RLT description for instructions with loops.

A microprocessor usually can be divided into two sections: the data processing part and the control part [2,11]. In this paper, we consider faults in both parts with the emphasis on the control faults.

### A. Data processing faults

(1) Data storage fault (R)/(R)'

This means that the content of register is changed from (R) to (R)' due to faults such as stuck-at and pattern sensitive faults.

(2) Data transfer fault +/+

The fault occurs in the transfer path between the sources and the destination. This type of fault includes stuck-at, bridging and pattern sensitive faults.

(3) Data manipulation fault (f)/(f)

This is the operation execution fault. Under this fault, the operation f is executed, but the result of operation is wrong.

### B. Control faults

This kind of fault involves register decoding faults, instruction decoding faults and other control faults. A register decoding fault means missing or changing the selected register, or selection of an extra register, denoted by  $R/\phi$ , R/R', and R/R+R' respectively. For instruction decoding faults, we consider that an instruction can be executed partially. It means missing or changing the selected operation, or selection of an extra operation in RTL. In this case, the instruction decoding fault may be  $I_j/\phi$ ,  $I_j/\Lambda I_j$ ,  $I_j/\Lambda I_k$ ,  $I_j/I_k$ ,  $I_j/I_j+\Lambda I_k$ ,  $I_j/I_j+I_k$ , and so forth, where  $\Lambda I$  means part of instruction I.

In fact, for most existing off-the-shelf microprocessors, the register decoding faults can be considered as instruction decoding faults, i.e.  $R/\phi c I_j/\phi$ ,  $R/R'c I_j/I_k$ ,  $R/R+R'c I_j/I_j+I_k$ , where  $I_j$ ,  $I_k$  are transfer instructions among registers. For example, let  $I_j$  be  $R_j$ -memory,  $I_k$  be  $R_k$ -memory, then fault  $R_j/R_k$  efault  $I_j/I_k$ . In addition, the control faults also include instruction execution sequence faults, condition faults and so on.

From the above observation, we assert that it is appropriate to represent the control faults at the RTL level. Therefore, we will define the above control faults at such a level. Let f denote  $D+f(S_1,S_2,...)$ , which is an operation on the instruction-under-test, and  $f\in\{f\}$ , where  $\{f\}$  is the set of RTL operations of a microprocessor. Let f' denote  $D'+f'(S_1',S_2',...)$ , which is an unexpected (faulty) operation, and  $f'\in\{f\}$ .

We now define three classes (i.e. nine subclasses F1, F2, ..., F9) of control faults as follows:

- (1) f/¢- No operation is executed.
  - **F1.** <u>f/</u>φ
- (2) f/f' Instead of performing the operation f, another operation f' is executed. It contains two subclasses of faults.
- F2.  $\underline{\delta f/f^{\dagger}}$ : Here  $\delta$  means that the destination registers D and D' are different and the fault is  $f/f^{\dagger}$ .
  - F3.  $\sigma f/f'$ :  $\sigma$  denotes that registers D and D' are the same.
- (3) f/f+f' In addition to operation f, another operation f' is also executed. It can be subdivided as follows.
  - (3a) Register D and D' are different.
- F4.  $\delta f/f+f'$ : The source register list of f and f' does not include D' and D respectively. We are not concerned with the execution order

of f and f'.

F5.  $\delta f/f'f$ : The source register list of f includes D'. f' is executed before performing operation f; i.e. the execution order is

1. 
$$D^{\dagger} \leftarrow f^{\dagger} (S_1^{\dagger}, S_2^{\dagger}, \ldots)$$

2. 
$$D \leftarrow f (S_1, S_2, \dots, D^**)$$

where register without \* denotes its content before executing the operation, register with \* denotes its content after executing the operation.

F6.  $\delta f/ff'$ : The source register list of f' includes D and the execution order is

1. Def 
$$(S_1, S_2, \ldots)$$

2. 
$$D^{\dagger} \leftarrow f^{\dagger} (S_1^{\dagger}, S_2^{\dagger}, ..., D^{\dagger})$$

- (3b) Registers D and D' are the same, i.e. D'=D. When the source register list of f and f' does not include D' and D respectively, if the execution order is f'f, the fault does not affect the execution of f. If the execution order is f', it is the same as the case with the fault of/f'.
- F7.  $\underline{\sigma f/f^{\dagger}f}$ : The source register list of f includes D, and the execution order is

1. 
$$D \leftarrow f^{\dagger} (S_1^{\dagger}, S_2^{\dagger}, ...)$$

2. D-f 
$$(S_1, S_2, ..., D*)$$

F8.  $\underline{\sigma f/ff'}$ : The source register list of f' includes D and the execution order is

1. Def 
$$(s_1, s_2, \ldots)$$

2. 
$$D \leftarrow f' (S_1', S_2', ..., D*)$$

F9. of/f/f': Both f and f' are executed at the same time.

Def 
$$(s_1, s_2, \ldots)$$

where L denotes logical AND or OR function. In this case, the final content of the destination D is the result of the composite value (i.e. the result of the

AND or OR function) of f and f.

Note that the above control faults can occur at any place in an instruction execution sequence. This control fault model can cover register deceding faults, instruction decoding faults (including partially instruction execution), instruction execution sequence faults, etc., since any control fault can always be defined as missing, changing, or extra PTL operations and will cause registers to have wrong contents.

### III: REQUIREMENTS FOR TESTING CONTROL FAULTS

Our purpose is to test the execution of microprocessor instructions.

Therefore, the objective of test pattern generation is to find the initial data in registers (test data) needed for testing functions of an instruction and also the sequence of instructions. This test data must satisfy certain requirements. From the control fault model given in Section II, we can obtain various requirements for testing control faults.

Let us establish the following notation. For a fault-free operation f, we have

 $V_{i}$  = the value of register i.

 $VS_i =$  the value of the operand in source register  $S_i$ .

VD = the value of the operand in destination register D.

VD\* = the value of destination register D storing the result of operation f.

For a faulty operation  $f^{\dagger}$ , we obtain  $VS_{i}^{\dagger}$ ,  $VD^{\dagger}$  and  $VD^{\dagger}*$  instead.

Theorem 1. Control faults  $T/\phi$ , T/T' and T/T+T' can be detected if the data values of registers satisfy the following requirements:

QTT1. 
$$V_i \neq V_j$$
,  $i \neq j$ 

QTT2. 
$$V_i L V_i \neq V_i$$
,  $i \neq j$ 

<u>Proof.</u> We shall prove this theorem by considering the nine fault classes defined in Section II.

- (i) For fault F1 (T/ $\phi$ ) and F2 ( $\delta$ T/T'), in order to verify transfer operation T, one needs  $\underline{VS} \neq \underline{VD}$ .
- (ii) For fault F3 (oT/T'), the results of T and T' should be different,
  i.e. VD\* # VD'\*. To obtain this result, we must have VS # VS'.
- (iii) For fault F4 ( $\delta T/T+T^{\dagger}$ ) and F5 ( $\delta T/T^{\dagger}T$ ), we need only to detect the extra operation  $T^{\dagger}$ . Therefore,  $\underline{VS^{\dagger} \neq VD^{\dagger}}$ .

- (iv) Fault F6 (6T/TT\*) means that transfer operation D S is performed first, then D\*+D\*. In order to detect the extra operation T\*, one needs VS ≠ VD\*.
- (v) Fault F7 (CT/T\*T) yields  $D \in S^*$  followed by  $D \in D^*$ . Therefore, we obtain the requirement  $VS^* \neq VD$ .
- (vi) For fault F8 ( $\sigma T/TT^{\dagger}$ ), we have D\*S, then D\*D\*. This fault does not affect operation T. In order to verify T, we need  $VS \neq VD$ .

The above six requirements belong to QTIL.

(vii) For fault F9 (°T/TLT'), the composite value of both results of T and T' should be different from the correct result of T. i.e.  $VSLVS' \neq VS$  which belongs to QTT2.

Q.E.D.

Theorem 2. Control faults T/A' and T/T+A' can be detected if the data values of registers satisfy the following requirements:

QTA1.  $V_{i} \neq V_{j}$  ,  $i \neq j$ 

QTA2.  $f_{\dot{L}}^{\dagger} \neq VS$ 

QTA3.  $f_{A}^{\dagger} \neq VD^{\dagger}$ 

QTA4.  $VSLi_{L}^{\dagger} \neq VS$ 

where  $f_A^{\dagger}$  is the result of operation of class A, i.e.  $f_A^{\dagger} = f_A^{\dagger}$  (VS<sub>1</sub>, VS<sub>2</sub>,...).

Proof. The proof is similar to the proof for Theorem 1. Since A' instead of T' is performed, we can change VS' to f' in the requirements, (ii), (iii), (v) and (vii) in the Proof for Theorem 1 to obtain the corresponding requirements for Theorem 2.

- (1) For F2 ( $\delta T/A^{\dagger}$ ),  $\underline{VS} \neq \underline{VD}$ , (QTA1).
- (ii) For F3 ( $\sigma T/A^{\dagger}$ ), VS  $\neq f_{A}^{\dagger}$ , (QTA2).
- (111) For F4 ( $\delta T/I + \Lambda^{\dagger}$ ) and F5 ( $\delta T/A^{\dagger}T$ ),  $f_{A}^{\dagger} \neq VD^{\dagger}$ , (QTA3).
- (iv) For F6 ( $\delta$ T/TA\*), it means that D+S first, then D\*+f\*<sub>A</sub> (S\*<sub>1</sub>,S\*<sub>2</sub>,...,D\*).

We need  $f_A^*$  (VS<sub>1</sub>,VS<sub>2</sub>,...,VS)  $\neq$  VD'. Since VS can be selected as any initial data value, it can be considered as one of several source operands. Therefore, we can rewrite  $f_A^*$  (VS<sub>1</sub>, VS<sub>2</sub>,...,VS)  $\neq$  VD' as  $f_A^*$   $\neq$  VD', (QTA3).

- (v) For F7 (cf/A'T),  $f_A^{\dagger} \neq VD^{\dagger}$ , (QTA3).
- (vi) For F8 (GT/TA'), it implies D+S followed by D- $f_A^*$  (S'\_1,S'\_2,...,D\*). This requires VS  $\neq f_A^*$  (VS',VS',...,VS). Here VS can be considered as a destination operand VD'. So we rewrite the inequality as VD'  $\neq f_A^*$ , (QTA3).
- (vii) For F9 ( $\sigma T/TLA^{\dagger}$ ),  $VSLf_{A}^{\dagger} = VS$ , (QTA4). Q.E.D. Theorem 3. Control faults  $A/\phi$ ,  $A/T^{\dagger}$  and  $A/A+T^{\dagger}$  can be detected if the data valued of registers satisfy the following inequalities.

QAT1.  $f_A \neq VD$ 

QAT2.  $f_A \neq VS^{\dagger}$ 

QAT3.  $V_i \neq V_i$ ,  $i \neq j$ 

QAT4.  $f_A \neq VD^{\dagger}$ 

QAT5.  $f_A$  (VS<sup>†</sup>)  $\neq$   $f_A$  (VD)

QATE.  $f_A L VS' \neq f_A$ 

where  $f_A = f_A (VS_1, VS_2, ...), f_A(VD) = f_A(VS_1, VS_2, ..., VD), f_A(VS') = f_A (VS_1, VS_2, ..., VS').$ 

<u>Proof.</u> Since arithmatical and logical operations instead of transfer operations are considered here, we can change VS to  $f_A$  in the cases (i), (ii), (iv), (vi) and (vii) of Theorem 1.

- (i) For F1 and F2,  $f_A \neq VD$ , (QAT1).
- (11) For F3,  $f_{A} \neq VS^{1}$ , (QAT2).
- (iii) For F4 and F5, VS' # VD', (QAT3).
- (iv) For F6,  $F_A \neq VD^{\dagger}$ , (QAT4).
- (v) F7 ( $\sigma$ A/T'A) means that D $\leftarrow$ S' followed by D $\leftarrow$ f<sub>A</sub>(S<sub>1</sub>,S<sub>2</sub>,...,D\*) which yields  $f_A(VS_1, VS_2,...,VS')$ . When there is no fault,  $f_A(VS_1, VS_2,...,VD)$  is obtained.

Thus the condition for detecting this fault is  $f_A(VS_1, VS_2,...,VS')$ #  $f_A(VS_1, VS_2,...,VD)$ , (QTA5).

(vi) For F8,  $f_A \neq VD$ , (QTA1).

(vii) For F9,  $f_A L VS' \neq f_A$ , (QTA6). Q.E.D.

Theorem 4. Control faults A/A' and A/A+A' can be detected if the data values of registers satisfy the following inequalities.

QAA1.  $f_A \neq VD$ 

QAA2.  $f_A \neq f_A^{\dagger}$ 

QAA3.  $f_A^{\dagger} \neq VD^{\dagger}$ 

QAA4.  $f_A^{\dagger}(f_A) \neq .VD^{\dagger}$ 

QAA5.  $f_A(f_A^{\dagger}) \neq f_A(VD)$ 

QAA6.  $f_A^{\dagger}$  ( $f_A$ )  $\neq$   $f_A$ 

QAA7.  $f_A L f_A^{\dagger} \neq f_A$ 

where  $f_A^{\dagger}(f_A) = f_A^{\dagger}(VS_1^{\dagger}, VS_2^{\dagger}, \dots, f_A), f_A^{\dagger}(f_A^{\dagger}) = f_A^{\dagger}(VS_1, VS_2, \dots, f_A^{\dagger}).$ 

Proof. For the same reason, we may change VS to  $f_A$ , and VS' to  $f_A'$  for cases (i) to (iii) and (vii) in Theorem 1 to obtain QAA1 to QAA3 and QAA7 respectively. In addition, since the results of A and A' may affect each other, we can obtain QAA4 to QAA6. Q.F.D.

Note that for requirement QAA7, if operation  $f_A$  and  $f_A^{\dagger}$  of class A are executed in the same unit (e.g. ALU), then both results of  $f_A$  and  $f_A^{\dagger}$  can no longer be considered as obtained separately. Instead, QAA7 may be considered as a data manipulation fault  $(f_A)/(f_A)^{\dagger}$ .

### IV. WRITE AND READ SEQUENCES

As a microprocessor is one type of sequential machine and all internal registers are memory elements of the sequential machine, the content of registers represents the state of the sequential machine. Therefore, the following procedure is utilized for testing microprocessors.

- 1. Initialization of state of registers.
- 2. Execution of the instruction-under-test.
- 3. Read the state of registers.

In fact, Steps 1 and 3 consist of write and read register sequences respectively.

Obviously, if we can guarantee the correctness of Steps 1 and 3 first, then

the testing problem will be simplified.

The testing approach used here is a kind of open loop testing [6]. It implies the use of a test equipment which provides the stimuli to the microprocessor and observes the responses from the microprocessor.

Now let us discuss write and read sequences which are used for writing and reading the register states of a microprocessor. They consist of several basic instructions, called the kernel of microprocessor. These instructions of the kernel can be carried out by a sequential machine, therefore, we can use a checking experiment to verify the kernel.

### A. The kernel of microprocessor

<u>Definition 1.</u> <u>Kernel instruction set</u> - A small subset of instructions of a microprocessor which can be used for constituting the write and read register sequences.

<u>Definition 2.</u> Register set - All internal registers of a microprocessor from the view of the architecture or programming.

<u>Definition 3.</u> <u>Kernel state</u> - The register state, i.e. certain set of data values of the registers.

<u>Definition 4.</u> <u>Kernel input</u> - The write sequence for writing a set of data into the registers, or the read sequence for reading out the contents of the registers.

<u>Definition 5.</u> <u>Kernel output</u> - The set of data values of the registers which are read out by the read sequence.

### B. Kernel instruction set

There exists many choices for the kernel instruction set. In order to keep the kernel small, the following requirements should be satisfied.

- 1. The number of instructions in the kernel instruction set should be small.
- 2. Functions of each kernel instruction should be as simple as possible. For instance, a kernel instruction contains mainly transfer type of operations, or small number of RTL operations.
- 3. In order to simplify addressing, the priority order of choosing the addressing mode of an instruction is as follows.
  - . For write register instructions: Immediate, Direct, Indirect.
  - . For read register instructions: Direct, Indirect.

For the existing off-the-shelf microprocessors, most registers can be written into or read from directly. These registers are called direct access registers. Others are indirect access registers which can be accessed through the direct access registers in certain order by using transfer instruction among registers.

### C. Kernel state

AND THE PARTY OF T

In order to simplify the testing, during the checking experiment, we only use a few states for the good kernel, i.e. we define several sets of data value for the registers. Therefore, we should choose the data values (test data) such that they can cover as many faults as possible.

### V. VERIFYING WRITE AND READ SEQUENCES

We use the checking experiment to verify the kernel of a microprocessor.

The main task is to decide how many states of the kernel and what test data we use.

Abraham and Parker [6] use the k-out-of-m codes for their "register read test" procedure, where m is the width of a code word (i.e. the length of register), and k is the number of 1's in the code word. As we will see, this type of code is powerful since it can be used as test data to cover most control faults by using fewer data. We will use the k-out-of-m codes for verifying the write and read sequences as well as for testing control faults. The k-out-of-m codes can also detect stuck-at type faults, but do not guarantee to cover all data processing faults.

To simplify the testing, we shall only use transfer operations of the kernel instructions in write and read sequences. Therefore, we only need to consider the requirements of Theorems 1 and 2.

### A. QTT1, QTA1 and QTT2

The k-out-of-m codes used as test data can satisfy requirements QTT1, QTA1 and QTT2 ( $V_i \neq V_j$  and  $V_i \perp V_j \neq V_1$ ). This is because in k-out-of-m codes, all code words are distinct and the AND(OR) operation of any two code words will reduce (increase) the number of 1's in the code word, thereby different from both original code words.

### B. QTA2

During the checking experiment for the kernel, there exists an input leaving the kernel state unchanged. Therefore, transfer operation (in write sequence) D-S implies VD = VS, then the requirement QTA2,  $f_A^* \neq VS$ , becomes  $f_A^* \neq VD$ . This is for detecting fault  $\sigma T/A^*$ , here D and D' are the same register. Therefore,

QTA2 is changed to  $f_A^* \neq VD^*$  which belongs to QTA3.

## C. QTA3

The requirement QTA3,  $f_A^{\dagger}(VS^{\dagger}, VS^{\dagger}, ...) \neq VD^{\dagger}$ , is for detecting an extra operation. We list the restrictions of operands (test data) for detecting operation of class A in Table 2.

| Extra Operations                                             | Restrictions of Operands                  |
|--------------------------------------------------------------|-------------------------------------------|
| BIT SET                                                      | ①                                         |
| BIT RESET                                                    | ①                                         |
| BIT COMPLEMENT                                               | No                                        |
| INCREMENT                                                    | No                                        |
| DECREMENT                                                    | No                                        |
| DECIMAL ADJUST                                               | 3                                         |
| SHIFT                                                        | # <u>0</u> (all 0s), 1 (all 1s)           |
| ROTATE                                                       | ≠ <u>0</u> , <u>1</u>                     |
| COMPLEMENT                                                   | No                                        |
| NEGATE                                                       | No                                        |
| CLEAR                                                        | ≠ <u>o</u>                                |
| ADDITION                                                     | ≠ <u>0</u>                                |
| ADDITION WITH CARRY                                          | ≠ <u>0</u> , -1                           |
| SUBTRACTION                                                  | ≠ <u>0</u>                                |
| SUBTRACTION WITH BORROW                                      | ≠ <u>0</u> , -1                           |
| AND                                                          | k-out-of-m codes                          |
| OR                                                           | k-out-of-m codes                          |
| xor                                                          | k-out-of-m codes                          |
| EXTEND SIGN                                                  | ≠ <u>0</u> , <u>1</u>                     |
| ADDITION, Des <sub>1</sub> + s <sub>2</sub>                  | the least significant bit LSE =1          |
| ADDITION, D+D + $s_1 + s_2$                                  | $\neq 0$ ; $vs_1' + vs_2' \neq 0$         |
| ADDITION, D+S <sub>1</sub> + S <sub>2</sub> + S <sub>3</sub> | (a) 1 2 1 1 2 1 1 1 1 1 1 1 1 1 1 1 1 1 1 |
| MULTIPLY                                                     | ≠ <u>0</u> , 1                            |
| DIVIDE                                                       | <u>≠ 0</u> , 1                            |
| BIT TEST                                                     | 2                                         |
| COMPARE                                                      | 2                                         |
| Modifying flags                                              | 3                                         |

Table 2. Restrictions of operands for detecting faulty arithmetic or logical operations

- ① If we use two tests in which the source operands are complements of each other, then one of the tests can detect the faulty operation.
- ② These operations only set or reset the flags. We use two tests with the identical source operands and two sets of flags which are complements to each other. The faulty will change one of the flags.
- 3 The execution of DECIMAL ADJUST (to add certain values) depends on the value of source operand and the flags. We can use either method in case 1 or either method in case 2 to detect the operation.
- 4 Operation D+S<sub>1</sub> + S<sub>2</sub> + S<sub>3</sub> is only used for memory address addition. Here D means external address bus. In this case, the unexpected operation does not affect the write and read registers. Hence it needs not to be considered for verifying the kernel.
- 5 During the checking experiment for the kernel, if we have considered the main operations in arithmetic and logical instructions as the unexpected operation, then we need not consider modifying flags which are auxiliary operations.

For other operations, the restrictions are obvious. For example,

ADDITION WITH CARRY, D'+D' + S' + CARRY, if CARRY = 0 with the restriction

VS' \( \neq 0 \) or CARRY' = 1 with the restriction is VS' \( \neq -1 \), then VD'\* \( \neq \text{VD'} \).

Since we use k-out-of-m codes as operands, these restrictions can be satisfied.

For operation D'+D' + S'\_1 + S'\_2, since the negative value of any k-out-of-m

code word will not be any k-out-of-m code, i.e. VS'\_1 + VS'\_2 \( \neq 0 \); so VD' + VS'\_1 + VS'\_2 \( \neq \text{VD'} \). For operation D'+S'\_1 + S'\_2, we can divide the k-out-of-m codes into two

groups with different LSB (Least Significant Bit). These two groups complement each other. The group with the restriction LSB = 1 will be used for testing.

For operation ROTATE, the restriction is for the case of the odd number of shifted bits. Otherwise, we need other restriction (e.g. using subset of k-out-of-m codes as operands).

### D. QTA4

The checking experiment does not guarantee requirement QTA4, VS L  $\mathbf{F}_{\mathbf{A}}^{\bullet} \neq \mathbf{VS}$ . For example, for the normal transfer operation T, D+S, with VS = 0101, VD = 0110 (using 2-out-of-4 codes), if there exists a fault  $\mathrm{CT}/\mathrm{TLA}^{\bullet}$  and the extra operation A' is INCREMENT, D+D+1, then  $\mathbf{f}_{\mathbf{A}}^{\bullet} = \mathrm{VD} + 1 = 0111$ . If L is an AND function, then VS L  $\mathbf{f}_{\mathbf{A}}^{\bullet} = \mathrm{VS}$ . Thus QTA4 cannot be satisfied. Therefore, we need another test procedure to remedy this. The remedy is to change the value of S to  $\underline{\mathbf{1}}$  or  $\underline{\mathbf{0}}$  depending upon L being AND or OR respectively. From the above example, we see that if L is an AND function, let VS =  $\underline{\mathbf{1}}$  then QTA4 becomes  $\mathbf{f}_{\mathbf{A}}^{\bullet} \neq \underline{\mathbf{1}}$  which can be satisfied. Similarly, if L is an OR function, let VS =  $\underline{\mathbf{0}}$  and QTA4 changes to  $\mathbf{f}_{\mathbf{A}}^{\bullet} \neq \underline{\mathbf{0}}$ .

Now let us check this remedy method for all operations of class A. Note that we only change the value of S in operation D+S, and D' in operation A' can be substituted by D.

- (i) For class Al, D-f\_A'(D). If we use k-out-of-m codes, the new requirement QTA4',  $f_A^{\dagger} \neq \underline{1}$  (0), depending on L, can be satisfied except the operation CLEAR with L being OR. But in this case, the result of  $f_A^{\dagger}$  is always 0 which does not affect the operation D+S.
- (ii) For class  $A_2$ ,  $D+f_A^*(D,S^*)$ . No matter  $S^*$  is the same register as S or not,  $QTA4^*$  is true except the following case: when  $S^*$  and S are the same register, then for  $QTA4^*$ ,  $f_A^* \neq 1$  ( $f_A^*$  is operation OR and L is AND) and  $f_A^* \neq 0$  ( $f_A^*$  is operation AND and L is OR) cannot be met. But in this case, if  $S^*$  and S are the same,  $VSLf_A^*$  is always the same as VS, the fault does not affect D+S, i.e. VSL(VD)VS = VS and VSV(VDAVS) = VS for any operands D and S.

- (iii) For the EXTEND SIGN operation, D+ $f_A^{\dagger}(S^{\dagger})$ , the result of  $f_A^{\dagger}$  is  $\underline{0}$  or  $\underline{1}$ . Similarly with case (1), either QTA4 can be met or the fault does not affect operation T.
- (iv) For classes A3 and AF, QTA4 can be satisfied. For the same reason as QTA3, we need to consider neither modifying flags operation nor  $D+S_1+S_2+S_3$ .
- (v) For MULTIPLY and DIVIDE, since the execution period of these operations generally is longer than that of transfer operations, fault  $\sigma f/fLf^{\dagger}$  cannot exist and we do not consider QTA4.

In summary, during the checking experiment we only need three sets of test data for internal registers. Let n be the number of the internal register, m be the length of registers. Suppose that m is even. Let  $\underline{V}$  denote a complete set of k-out-of-m code words. Usually,  $k = \frac{m}{2}$ .  $|\underline{V}| = (\frac{m}{m/2})$  will be the maximum number of distinct codes. We divide them into two groups,  $\underline{V}_0$  and  $\underline{V}_1$  with different LSB. These two groups are complementary to each other. Note that for most microprocessors, m is even, and  $|\underline{V}| > 2n$ .

Let  $\{\alpha\} = \{\alpha_1, \alpha_2, \dots, \alpha_n\}$ ,  $\{\overline{\alpha}\} = \overline{\alpha}_1, \overline{\alpha}_2, \dots, \overline{\alpha}_n\}$ ,  $\{\alpha\}$  and  $\{\overline{\alpha}\}$  belongs to  $\underline{V}_0$  and  $\underline{V}_1$  respectively. We now construct four sets of data as follows.

| Flag register         | Other registers                                                        |  |
|-----------------------|------------------------------------------------------------------------|--|
| α <sub>1</sub>        | α <sub>2</sub> , α <sub>3</sub> ,, α <sub>n</sub>                      |  |
| $\overline{\alpha}_1$ | $\overline{\alpha}_2$ , $\overline{\alpha}_3$ ,, $\overline{\alpha}_n$ |  |
| α <sub>1</sub>        | $\overline{\alpha}_2$ , $\overline{\alpha}_3$ ,, $\overline{\alpha}_n$ |  |
| <sup>α</sup> 1        | $\alpha_2, \alpha_3, \ldots, \alpha_n$                                 |  |

In order to satisfy requirement QTA3, we can choose any three sets of data as the initial values of registers (test data). In fact, one needs a few more data as external bus inputs during testing. Therefore, the final number of code words in  $\{\alpha\}$  and  $\{\overline{\alpha}\}$  is larger than n.

It should be pointed out that the program counter (PC) is easy to test.

We can put a direct addressing branch instruction at the end of a write sequence.

This instruction stores a particular value into PC, then the content of PC is checked at the beginning of the read sequence by observing the address bus.

Now we will derive the checking sequence for the kernel. As we mentioned before, we only use three sets of test data for the kernel, namely a, b and c. First of all, just like a sequential machine we have to obtain a flow table of the kernel. We consider two cases.

<u>Case 1.</u> For a microprocessor without indirect access registers, we obtain the following flow table.

|   | Wa  | w <sub>b</sub> | W <sub>C</sub> | R             |
|---|-----|----------------|----------------|---------------|
| A | ① A | (4) B          | ⑦ c            | () A,a        |
| В | ② A | (5) B          | <b>®</b> с     | <b>О</b> в, ь |
| С | 3 A | 6 B            | <b>9</b> с     | Ø C,c         |

<u>Case 2</u>. For a microprocessor with indirect access registers, we obtain the following table.

|    | ¥<br>a     | w <sub>b</sub> | W <sub>C</sub> | R             |
|----|------------|----------------|----------------|---------------|
| A  | ① A        | (7) B          | OB c           | ① A*,a        |
| В  | ② A        | <b>8</b> B     | <b></b> € C    | <b>⊘</b> B*,b |
| С  | 3 A        | (9) B          | <b>©</b> c     | ②) C*,c       |
| A* | <b>4</b> A | <b>(</b> ) B   | € c            | ② A*,a*       |
| В* | (5) A      | <b>()</b> B    | Ø c            | ② B*,b*       |
| C* | 6 A        | <b>()</b> B    | € C            | @ C*,c*       |

where  $W_a$ ,  $W_b$ ,  $W_c$ - write sequences for writing a, b and c respectively.

R - read sequence.

A, B, C, A\*, B\*, C\* - Kernel states. A, B and C are the states after applying the corresponding write sequences  $W_a$ ,  $W_b$ ,  $W_c$ . A\*, B\* and C\* are the states after applying read sequence R.

a, b, c, a\*, b\*, c\* - Kernel output sequences produced by read sequence R for states A, B, C, A\*, B\*, C\* respectively.

(1) denote the state transition i.

Since we only use three sets of test data, the number of states of the kernel is constant. It means that the above flow tables are independent of microprocessors. Therefore, we can easily obtain the checking sequences with the same form.

There are three requirements to derive a checking sequence [12,13].

- 1. Initialization of the machine (kernel) state using sychronizing sequence or homing sequence.
  - 2. Identify all machine states using distinguishing sequence.
  - 3. Verify each transition using distinguishing sequence.

For our kernel, there exists sychronizing sequences  $W_a$  or  $W_b$  or  $W_c$  and distinguishing sequence R. Therefore, we can easily derive checking sequences as follows.

### Checking Sequence 1 (for case 1)

### Checking sequence 2 (for case 2)





Finally, we can obtain two test procedures for verifying the write and read sequences as follows.

### Procedure 1: Checking experiment

- If a microprocessor does not have indirect access registers, we use checking sequence 1.
- 2. If a microprocessor has indirect access registers, we use checking sequence 2.

### Procedure 2: Remedy testing

For each of the three sets of register values a, b and c, do the following for each register.

- 1. Initialization of all registers.
- 2. Write  $\underline{1}$  or  $\underline{0}$  into the given register depending on the circuit implementation.
- 3. Read the given register.

Therefore, from the previous discussion in this section, we obtain the following theorem.

Theorem 5. Procedures 1 and 2 can verify write and read sequences, and after that the registers of a microprocessor can be initialized to any values.

### VI. TESTING CONTROL FAULTS

During the verification of the correctness of write and read sequences, we are only concerned with certain transfer operations (not all RTL operations) in the kernel instructions. Therefore, when we test instruction decoding and other control faults, we need to test all instructions included in the kernel.

Procedure 3. Testing instruction decoding and other control faults

For each instruction, do the following steps,

- Initialize register state using any particular initial values (test data).
- 2. Execute the instruction instruction-under test.
- 3. Read register state.

Note that we should first try to use three sets of data values a, b and c at Step 1.

Generally we need several tests for each instruction to detect the instruction decoding and other control faults. Obviously, the lower bound of the number of tests using Procedure 3 for each instruction is two. This is because any kind of microprocessors has several pairs of conditional branch instructions based on two different values for the same condition source. Therefore, when any instruction is under test, in order to detect an unexpected branch instruction due to a fault, we need at least two test patterns.

The upper bound on the test for each instruction is dependent upon the microprocessor-under-test. We can roughly estimate the order of tests for detecting instruction decoding faults. We consider  $n_{\tilde{I}}$  instructions to be tested, assume that each instruction corresponds to an operation, class T or class A, used for distinguishing instructions from each other. Let  $n_{\tilde{I}\tilde{I}}$  and  $n_{\tilde{I}\tilde{A}}$  denote the number of instructions which have operations class T and

class A respectively, i.e.  $n_T = n_{TT} + n_{TA}$ .

### A. Testing instruction class T

### 1. The case of Theorem 1

Since for some condition branch instructions, their operations belong to class T, and they are condition transfer operations. In order to detect this unexpected operation  $T^{\dagger}$ , two tests, in which test data are complement to each other, are sufficient.

### 2. The case of Theorem 2

First of all, we use three sets of initial values a, b and c for covering the requirements QTA1 and QTA3. In order to satisfy QTA2 and QTA4, we can modify a, b and c separately as new test data. As we have discussed in Section V, if the instruction-under-test has a transfer operation D+S, we can change VS in original data a, b and c to VD, then QTA2 becomes QTA3 which can be satisfied. Similarly, we can change VS to  $\underline{1}$  (or  $\underline{0}$ ) to satisfy QTA4. Here we need nine tests altogether. Thus, the order of the number of tests for testing instruction class  $\underline{T}$  is  $0(n_{\underline{T}\underline{T}})$ .

### B. Testing instruction class A

### 1. The case of Theorem 3

Test data a, b and c can cover QAT1 and QAT3. Similarly, in order to satisfy QAT2, QAT4, QAT5 and QAT6, we can modify a, b and c in turn. First, we change VS' and VD' to VD for covering QAT2 and QAT4 respectively. These changes are done for each register, so it needs 3n tests, where n is the number of registers. Then we change VS' to a particular value for covering QAT5 and QAT6 separately. It needs 6n tests. So the total number of tests for each instruction in this class will be 9n+3.

### 2. The case of Theorem 4

We need three tests using a, b and c for covering QAA1 and QAA2.

Then we attempt to find five particular tests to satisfy QAA3 through QAA7 for each unexpected operation A'. So the number of tests for each instruction in this class will be 3 + 5 ( $n_{IA} - 1$ ) = 5  $n_{IA} - 2$ .

Therefore, the order of the number of tests for testing instruction class A is 0 ( $n \cdot n_{IA} + n_{IA}^2$ ).

### VII. CONCLUSION

In this paper, we presented three test procedures to verify the write and read sequences and to test control faults at the RTL level.

A control fault model is defined at the RTL level instead of the instruction level. This allows us to model the fault more accurately.

Based on the control fault model, we consider the write and read register sequence functions as the kernel of a microprocessor independent of microprocessors. The similarity between the kernel of a microprocessor and a sequential machine allows us to obtain the checking sequences to verify the kernel. Therefore, testing of microprocessors can be simplified.

We mainly use k-out-of-m codes as test data which can cover more control faults. Therefore, the number of test patterns can be reduced.

Further work includes the enumeration of the control faults at the RTL level for the generation of tests for covering all possible control faults in a microprocessor.

### ACKNOWLEDGEMENT

The authors thank Dr. K.K. Saluja for his helpful comments and suggestions on this paper.

ではないない。

### VII. CONCLUSION

In this paper, we presented three test procedures to verify the write and read sequences and to test control faults at the RTL level.

A control fault model is defined at the RTL level instead of the instruction level. This allows us to model the fault more accurately.

Based on the control fault model, we consider the write and read register sequence functions as the kernel of a microprocessor independent of microprocessors. The similarity between the kernel of a microprocessor and a sequential machine allows us to obtain the checking sequences to verify the kernel. Therefore, testing of microprocessors can be simplified.

We mainly use k-out-of-m codes as test data which can cover more control faults. Therefore, the number of test patterns can be reduced.

Further work includes the enumeration of the control faults at the RTL level for the generation of tests for covering all possible control faults in a microprocessor.

### **ACKNOWLEDGEMENT**

The authors thank Dr. K.K. Saluja for his helpful comments and suggestions on this paper.

### VI. TESTING CONTROL FAULTS

During the verification of the correctness of write and read sequences, we are only concerned with certain transfer operations (not all RTL operations) in the kernel instructions. Therefore, when we test instruction decoding and other control faults, we need to test all instructions included in the kernel.

Procedure 3. Testing instruction decoding and other control faults

For each instruction, do the following steps,

- Initialize register state using any particular initial values (test data).
- 2. Execute the instruction instruction-under test.
- 3. Read register state.

Note that we should first try to use three sets of data values a, b and c at Step 1.

Generally we need several tests for each instruction to detect the instruction decoding and other control faults. Obviously, the lower bound of the number of tests using Procedure 3 for each instruction is two. This is because any kind of microprocessors has several pairs of conditional branch instructions based on two different values for the same condition source. Therefore, when any instruction is under test, in order to detect an unexpected branch instruction due to a fault, we need at least two test patterns.

The upper bound on the test for each instruction is dependent upon the microprocessor-under-test. We can roughly estimate the order of tests for detecting instruction decoding faults. We consider  $n_{\rm I}$  instructions to be tested, assume that each instruction corresponds to an operation, class T or class A, used for distinguishing instructions from each other. Let  $n_{\rm II}$  and  $n_{\rm IA}$  denote the number of instructions which have operations class T and

class A respectively, i.e.  $n_I = n_{II} + n_{IA}$ .

### A. Testing instruction class T

### 1. The case of Theorem 1

Since for some condition branch instructions, their operations belong to class T, and they are condition transfer operations. In order to detect this unexpected operation T', two tests, in which test data are complement to each other, are sufficient.

### 2. The case of Theorem 2

First of all, we use three sets of initial values a, b and c for covering the requirements QTA1 and QTA3. In order to satisfy QTA2 and QTA4, we can modify a, b and c separately as new test data. As we have discussed in Section V, if the instruction-under-test has a transfer operation D+S, we can change VS in original data a, b and c to VD, then QTA2 becomes QTA3 which can be satisfied. Similarly, we can change VS to  $\underline{1}$  (or  $\underline{0}$ ) to satisfy QTA4. Here we need nine tests altogether. Thus, the order of the number of tests for testing instruction class  $\underline{T}$  is  $0(n_{\underline{T}\underline{T}})$ .

### B. Testing instruction class A

### 1. The case of Theorem 3

Test data a, b and c can cover QAT1 and QAT3. Similarly, in order to satisfy QAT2, QAT4, QAT5 and QAT6, we can modify a, b and c in turn. First, we change VS' and VD' to VD for covering QAT2 and QAT4 respectively. These changes are done for each register, so it needs 3n tests, where n is the number of registers. Then we change VS' to a particular value for covering QAT5 and QAT6 separately. It needs 6n tests. So the total number of tests for each instruction in this class will be 9n+3.

### 2. The case of Theorem 4

We need three tests using a, b and c for covering QAA1 and QAA2.

Then we attempt to find five particular tests to satisfy QAA3 through QAA7 for each unexpected operation  $A^{\dagger}$ . So the number of tests for each instruction in this class will be 3 + 5 ( $n_{IA}$  -1) = 5  $n_{IA}$  -2.

Therefore, the order of the number of tests for testing instruction class A is 0  $(n \cdot n_{IA} + n_{IA}^2)$ .

### VII. CONCLUSION

In this paper, we presented three test procedures to verify the write and read sequences and to test control faults at the RTL level.

A control fault model is defined at the RTL level instead of the instruction level. This allows us to model the fault more accurately.

Based on the control fault model, we consider the write and read register sequence functions as the kernel of a microprocessor independent of microprocessors. The similarity between the kernel of a microprocessor and a sequential machine allows us to obtain the checking sequences to verify the kernel. Therefore, testing of microprocessors can be simplified.

We mainly use k-out-of-m codes as test data which can cover more control faults. Therefore, the number of test patterns can be reduced.

Further work includes the enumeration of the control faults at the RTL level for the generation of tests for covering all possible control faults in a microprocessor.

### ACKNOWLEDGEMENT

The authors thank Dr. K.K. Saluja for his helpful comments and suggestions on this paper.

### REFERENCES

- [1] A.C.L. Chiang and R. McCaskill, "Two New Approaches Simplify Testing of Microprocessors", <u>Electronics</u>, 22 Jan. 1976, p. 100.
- [2] S.M. Thatte and J.A. Abraham, "Methodology for Functional Level Testing of Microprocessors", 8th International Symposium on Fault-Telerant Computing, Toulouse, France, June 1978, pp. 90-95.
- [3] S.M. Thatte and J.A. Abraham, "Test Generation for Microprocessors", IEEE Trans. on Computers, C-29, No. 6, June 1980, pp. 429-441.
- [4] S.M. Thatte, "Test Generation for Microprocessors", Coordinated Science Laboratory Report R-842, University of Illinois at Urbana-Champaign, May 1979.
- [5] J.A. Abraham and S.M. Thatte, "Fault Coverage of Testing Program for a Microprocessor", 1979 Test Conference, Oct. 1979, pp. 18-22.
- [6] J.A. Abraham and K.P. Parker, "Practical Microprocessor Testing: Open and Closed Loop Approaches", IEEE Compcon, Spring 1981, pp. 308-311.
- [7] Y. Min and S.Y.H. Su, "Testing Functional Faults in VLSI", 19th Design Automation Conference, Los Vegas, Nevada, 1982, pp. 384-392.
- [8] K.K. Saluja, L. Shen and S.Y.P. Su, "A Simplified Algorithm for Testing Microprocessors", 1983 Test Conference, Oct. 1983, pp. 668-675.
- [9] C. Robach and G. Saucier, "Microprocessor Punctional Testing", 1980 Test Conference, Nov. 1980, pp. 433-443.

ACCOUNT THE STATE OF THE PARTY OF THE PARTY

- [10] M.A. Annaratone and M.G. Sami, "An Approach to Functional Testing of Microprocessors", 12th International Symposium on Fault-Tolerant Computing, Santa Monica, CA, June 1982, pp. 158-164.
- [11] B. Courtois, "Functional Testing of the Control Section of Integrated Processing Units", RR No. 203, IMAG, Univ. of Grenoble, France, May 1980.
- [12] F.C. Hennie, Finite-State Models for Logical Machines, John Wiley & Sons, Inc., NY, 1968.
- [13] A.D. Friedman and D. Menon, <u>Fault Detection in Digital Circuits</u>, Prentice-Hall, NJ, 1971.

## FILMED

1-86

DTIC